1 package org.infinispan.configuration.cache;
2
3 import static org.infinispan.configuration.cache.TransactionConfiguration.*;
4
5 import java.lang.invoke.MethodHandles;
6 import java.util.concurrent.TimeUnit;
7
8 import javax.transaction.Synchronization;
9 import javax.transaction.TransactionManager;
10 import javax.transaction.xa.XAResource;
11
12 import org.infinispan.commons.configuration.Builder;
13 import org.infinispan.commons.configuration.attributes.Attribute;
14 import org.infinispan.commons.configuration.attributes.AttributeSet;
15 import org.infinispan.configuration.global.GlobalConfiguration;
16 import org.infinispan.transaction.LockingMode;
17 import org.infinispan.transaction.TransactionMode;
18 import org.infinispan.transaction.TransactionProtocol;
19 import org.infinispan.transaction.lookup.TransactionManagerLookup;
20 import org.infinispan.transaction.lookup.TransactionSynchronizationRegistryLookup;
21 import org.infinispan.util.logging.Log;
22 import org.infinispan.util.logging.LogFactory;
23
24
25
26
27
28
29
30 public class TransactionConfigurationBuilder extends AbstractConfigurationChildBuilder implements Builder<TransactionConfiguration> {
31 private static final Log log = LogFactory.getLog(MethodHandles.lookup().lookupClass());
32 private final AttributeSet attributes;
33 private final RecoveryConfigurationBuilder recovery;
34
35 TransactionConfigurationBuilder(ConfigurationBuilder builder) {
36 super(builder);
37 attributes = TransactionConfiguration.attributeDefinitionSet();
38 this.recovery = new RecoveryConfigurationBuilder(this);
39 }
40
41
42
43
44
45
46
47 public TransactionConfigurationBuilder autoCommit(boolean b) {
48 attributes.attribute(AUTO_COMMIT).set(b);
49 return this;
50 }
51
52
53
54
55
56
57
58
59
60
61 public TransactionConfigurationBuilder cacheStopTimeout(long l) {
62 attributes.attribute(CACHE_STOP_TIMEOUT).set(l);
63 return this;
64 }
65
66
67
68
69
70
71
72
73
74
75 public TransactionConfigurationBuilder cacheStopTimeout(long l, TimeUnit unit) {
76 return cacheStopTimeout(unit.toMillis(l));
77 }
78
79
80
81
82
83
84
85
86
87
88
89 @Deprecated
90 public TransactionConfigurationBuilder eagerLockingSingleNode(boolean b) {
91 attributes.attribute(EAGER_LOCKING_SINGLE_NODE).set(b);
92 return this;
93 }
94
95
96
97
98
99
100
101 public TransactionConfigurationBuilder lockingMode(LockingMode lockingMode) {
102 attributes.attribute(LOCKING_MODE).set(lockingMode);
103 return this;
104 }
105
106 LockingMode lockingMode() {
107 return attributes.attribute(LOCKING_MODE).get();
108 }
109
110
111
112
113
114
115
116
117 public TransactionConfigurationBuilder syncCommitPhase(boolean b) {
118 attributes.attribute(SYNC_COMMIT_PHASE).set(b);
119 return this;
120 }
121
122
123
124
125
126
127 boolean syncCommitPhase() {
128 return attributes.attribute(SYNC_COMMIT_PHASE).get();
129 }
130
131
132
133
134
135
136
137
138
139 public TransactionConfigurationBuilder syncRollbackPhase(boolean b) {
140 attributes.attribute(SYNC_ROLLBACK_PHASE).set(b);
141 return this;
142 }
143
144
145
146
147
148 public TransactionConfigurationBuilder transactionManagerLookup(TransactionManagerLookup tml) {
149 attributes.attribute(TRANSACTION_MANAGER_LOOKUP).set(tml);
150 if (tml != null) {
151 this.transactionMode(TransactionMode.TRANSACTIONAL);
152 }
153 return this;
154 }
155
156
157
158
159
160 public TransactionConfigurationBuilder transactionSynchronizationRegistryLookup(TransactionSynchronizationRegistryLookup lookup) {
161 attributes.attribute(TRANSACTION_SYNCHRONIZATION_REGISTRY_LOOKUP).set(lookup);
162 return this;
163 }
164
165 public TransactionConfigurationBuilder transactionMode(TransactionMode transactionMode) {
166 attributes.attribute(TRANSACTION_MODE).set(transactionMode);
167 return this;
168 }
169
170 TransactionMode transactionMode() {
171 if (attributes.attribute(TRANSACTION_MODE).isModified()) {
172 return attributes.attribute(TRANSACTION_MODE).get();
173 } else {
174 return null;
175 }
176 }
177
178
179
180
181
182
183
184
185
186
187
188 @Deprecated
189 public TransactionConfigurationBuilder useEagerLocking(boolean b) {
190 this.attributes.attribute(USE_EAGER_LOCKING).set(b);
191 return this;
192 }
193
194
195
196
197
198
199
200 public TransactionConfigurationBuilder useSynchronization(boolean b) {
201 attributes.attribute(USE_SYNCHRONIZATION).set(b);
202 return this;
203 }
204
205
206
207
208
209
210 boolean useSynchronization() {
211 return attributes.attribute(USE_SYNCHRONIZATION).get();
212 }
213
214
215
216
217
218
219 public RecoveryConfigurationBuilder recovery() {
220 return recovery;
221 }
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236 public TransactionConfigurationBuilder use1PcForAutoCommitTransactions(boolean b) {
237 attributes.attribute(USE_1_PC_FOR_AUTO_COMMIT_TRANSACTIONS).set(b);
238 return this;
239 }
240
241
242
243
244 public TransactionConfigurationBuilder reaperWakeUpInterval(long interval) {
245 attributes.attribute(REAPER_WAKE_UP_INTERVAL).set(interval);
246 return this;
247 }
248
249
250
251
252 public TransactionConfigurationBuilder completedTxTimeout(long timeout) {
253 attributes.attribute(COMPLETED_TX_TIMEOUT).set(timeout);
254 return this;
255 }
256
257 public TransactionConfigurationBuilder transactionProtocol(TransactionProtocol transactionProtocol) {
258 attributes.attribute(TRANSACTION_PROTOCOL).set(transactionProtocol);
259 return this;
260 }
261
262 @Override
263 public void validate() {
264 Attribute<Long> reaperWakeUpInterval = attributes.attribute(REAPER_WAKE_UP_INTERVAL);
265 Attribute<Long> completedTxTimeout = attributes.attribute(COMPLETED_TX_TIMEOUT);
266 if (reaperWakeUpInterval.get()< 0)
267 throw log.invalidReaperWakeUpInterval(reaperWakeUpInterval.get());
268 if (completedTxTimeout.get() < 0)
269 throw log.invalidCompletedTxTimeout(completedTxTimeout.get());
270 if(attributes.attribute(TRANSACTION_PROTOCOL).get() == TransactionProtocol.TOTAL_ORDER) {
271
272 if(transactionMode() != TransactionMode.TRANSACTIONAL) {
273 throw log.invalidTxModeForTotalOrder(transactionMode());
274 }
275
276
277 if(!clustering().cacheMode().isReplicated() && !clustering().cacheMode().isDistributed()) {
278 throw log.invalidCacheModeForTotalOrder(clustering().cacheMode().friendlyCacheModeString());
279 }
280
281 if (recovery.create().enabled()) {
282 throw log.unavailableTotalOrderWithTxRecovery();
283 }
284
285 if (lockingMode() != LockingMode.OPTIMISTIC) {
286 throw log.invalidLockingModeForTotalOrder(lockingMode());
287 }
288 }
289 recovery.validate();
290 }
291
292 @Override
293 public void validate(GlobalConfiguration globalConfig) {
294 recovery.validate(globalConfig);
295 }
296
297 @Override
298 public TransactionConfiguration create() {
299 if (attributes.attribute(USE_EAGER_LOCKING).get()) {
300 lockingMode(LockingMode.PESSIMISTIC);
301 }
302 if (transactionMode() == null && getBuilder().invocationBatching().isEnabled())
303 transactionMode(TransactionMode.TRANSACTIONAL);
304 else if (transactionMode() == null)
305 transactionMode(TransactionMode.NON_TRANSACTIONAL);
306 return new TransactionConfiguration(attributes.protect(), recovery.create());
307 }
308
309 @Override
310 public TransactionConfigurationBuilder read(TransactionConfiguration template) {
311 this.attributes.read(template.attributes());
312 this.recovery.read(template.recovery());
313
314 return this;
315 }
316
317 @Override
318 public String toString() {
319 return "TransactionConfigurationBuilder [attributes=" + attributes + ", recovery=" + recovery + "]";
320 }
321
322 }